//
// VMime library (http://www.vmime.org)
// Copyright (C) 2002 Vincent Richard <vincent@vmime.org>
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License as
// published by the Free Software Foundation; either version 3 of
// the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// General Public License for more details.
//
// You should have received a copy of the GNU General Public License along
// with this program; if not, write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
// Linking this library statically or dynamically with other modules is making
// a combined work based on this library.  Thus, the terms and conditions of
// the GNU General Public License cover the whole combination.
//

#include "vmime/config.hpp"


#if VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_SMTP


#include "vmime/net/smtp/SMTPResponse.hpp"

#include "vmime/platform.hpp"
#include "vmime/utility/stringUtils.hpp"

#include "vmime/net/socket.hpp"
#include "vmime/net/timeoutHandler.hpp"
#include "vmime/net/tracer.hpp"

#include <cctype>


namespace vmime {
namespace net {
namespace smtp {


SMTPResponse::SMTPResponse(
	const shared_ptr <tracer>& tr,
	const shared_ptr <socket>& sok,
	const shared_ptr <timeoutHandler>& toh,
	const state& st
)
	: m_socket(sok),
	  m_timeoutHandler(toh),
	  m_tracer(tr),
	  m_responseBuffer(st.responseBuffer),
	  m_responseContinues(false) {

}


SMTPResponse::SMTPResponse(const SMTPResponse&)
	: vmime::object() {

	// Not used
}


int SMTPResponse::getCode() const {

	const int firstCode = m_lines[0].getCode();

	for (unsigned int i = 1 ; i < m_lines.size() ; ++i) {

		// All response codes returned must be equal
		// or else this in an error...
		if (m_lines[i].getCode() != firstCode) {
			return 0;
		}
	}

	return firstCode;
}


const SMTPResponse::enhancedStatusCode SMTPResponse::getEnhancedCode() const {

	return m_lines[m_lines.size() - 1].getEnhancedCode();
}


const string SMTPResponse::getText() const {

	string text = m_lines[0].getText();

	for (unsigned int i = 1 ; i < m_lines.size() ; ++i) {

		text += '\n';
		text += m_lines[i].getText();
	}

	return text;
}


// static
shared_ptr <SMTPResponse> SMTPResponse::readResponse(
	const shared_ptr <tracer>& tr,
	const shared_ptr <socket>& sok,
	const shared_ptr <timeoutHandler>& toh,
	const state& st
) {

	shared_ptr <SMTPResponse> resp =
		shared_ptr <SMTPResponse>(new SMTPResponse(tr, sok, toh, st));

	resp->readResponse();

	return resp;
}


void SMTPResponse::readResponse() {

	responseLine line = getNextResponse();
	m_lines.push_back(line);

	while (m_responseContinues) {
		line = getNextResponse();
		m_lines.push_back(line);
	}
}


const string SMTPResponse::readResponseLine() {

	string currentBuffer = m_responseBuffer;

	if (m_timeoutHandler) {
		m_timeoutHandler->resetTimeOut();
	}

	while (true) {

		// Get a line from the response buffer
		const size_t lineEnd = currentBuffer.find_first_of('\n');

		if (lineEnd != string::npos) {

			size_t actualLineEnd = lineEnd;

			if (actualLineEnd != 0 && currentBuffer[actualLineEnd - 1] == '\r') {  // CRLF case
				actualLineEnd--;
			}

			const string line(currentBuffer.begin(), currentBuffer.begin() + actualLineEnd);

			currentBuffer.erase(currentBuffer.begin(), currentBuffer.begin() + lineEnd + 1);
			m_responseBuffer = currentBuffer;

			if (m_tracer) {
				m_tracer->traceReceive(line);
			}

			return line;
		}

		// Check whether the time-out delay is elapsed
		if (m_timeoutHandler && m_timeoutHandler->isTimeOut()) {

			if (!m_timeoutHandler->handleTimeOut()) {
				throw exceptions::operation_timed_out();
			}

			m_timeoutHandler->resetTimeOut();
		}

		// Receive data from the socket
		string receiveBuffer;
		m_socket->receive(receiveBuffer);

		if (receiveBuffer.empty()) {   // buffer is empty
			m_socket->waitForRead();
			continue;
		}

		currentBuffer += receiveBuffer;
	}
}


const SMTPResponse::responseLine SMTPResponse::getNextResponse() {

	string line = readResponseLine();

	const int code = extractResponseCode(line);
	string text;

	m_responseContinues = (line.length() >= 4 && line[3] == '-');

	if (line.length() > 4) {
		text = utility::stringUtils::trim(line.substr(4));
	} else {
		text = "";
	}

	return responseLine(code, text, extractEnhancedCode(text));
}


// static
int SMTPResponse::extractResponseCode(const string& response) {

	int code = 0;

	if (response.length() >= 3) {

		code = (response[0] - '0') * 100
		     + (response[1] - '0') * 10
		     + (response[2] - '0');
	}

	return code;
}


// static
const SMTPResponse::enhancedStatusCode SMTPResponse::extractEnhancedCode(const string& responseText) {

	enhancedStatusCode enhCode;

	std::istringstream iss(responseText);
	iss.imbue(std::locale::classic());

	if (std::isdigit(iss.peek())) {

		iss >> enhCode.klass;

		if (iss.get() == '.' && std::isdigit(iss.peek())) {

			iss >> enhCode.subject;

			if (iss.get() == '.' && std::isdigit(iss.peek())) {

				iss >> enhCode.detail;
				return enhCode;
			}
		}
	}

	return enhancedStatusCode();   // no enhanced code found
}


const SMTPResponse::responseLine SMTPResponse::getLineAt(const size_t pos) const {

	return m_lines[pos];
}


size_t SMTPResponse::getLineCount() const {

	return m_lines.size();
}


const SMTPResponse::responseLine SMTPResponse::getLastLine() const {

	return m_lines[m_lines.size() - 1];
}


const SMTPResponse::state SMTPResponse::getCurrentState() const {

	state st;
	st.responseBuffer = m_responseBuffer;

	return st;
}



// SMTPResponse::responseLine

SMTPResponse::responseLine::responseLine(
	const int code,
	const string& text,
	const enhancedStatusCode& enhCode
)
	: m_code(code),
	  m_text(text),
	  m_enhCode(enhCode) {

}


void SMTPResponse::responseLine::setCode(const int code) {

	m_code = code;
}


int SMTPResponse::responseLine::getCode() const {

	return m_code;
}


void SMTPResponse::responseLine::setEnhancedCode(const enhancedStatusCode& enhCode) {

	m_enhCode = enhCode;
}


const SMTPResponse::enhancedStatusCode SMTPResponse::responseLine::getEnhancedCode() const {

	return m_enhCode;
}


void SMTPResponse::responseLine::setText(const string& text) {

	m_text = text;
}


const string SMTPResponse::responseLine::getText() const {

	return m_text;
}



// SMTPResponse::enhancedStatusCode


SMTPResponse::enhancedStatusCode::enhancedStatusCode()
	: klass(0),
	  subject(0),
	  detail(0) {

}


SMTPResponse::enhancedStatusCode::enhancedStatusCode(const enhancedStatusCode& enhCode)
	: klass(enhCode.klass),
	  subject(enhCode.subject),
	  detail(enhCode.detail) {

}


std::ostream& operator<<(std::ostream& os, const SMTPResponse::enhancedStatusCode& code) {

	os << code.klass << '.' << code.subject << '.' << code.detail;
	return os;
}


} // smtp
} // net
} // vmime


#endif // VMIME_HAVE_MESSAGING_FEATURES && VMIME_HAVE_MESSAGING_PROTO_SMTP

